﻿<!DOCTYPE html>
<html>
<head>
<title>第6章：快速响应的用户界面</title>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<style type="text/css">
/* GitHub stylesheet for MarkdownPad (http://markdownpad.com) */
/* Author: Nicolas Hery - http://nicolashery.com */
/* Version: b13fe65ca28d2e568c6ed5d7f06581183df8f2ff */
/* Source: https://github.com/nicolahery/markdownpad-github */

/* RESET
=============================================================================*/

html, body, div, span, applet, object, iframe, h1, h2, h3, h4, h5, h6, p, blockquote, pre, a, abbr, acronym, address, big, cite, code, del, dfn, em, img, ins, kbd, q, s, samp, small, strike, strong, sub, sup, tt, var, b, u, i, center, dl, dt, dd, ol, ul, li, fieldset, form, label, legend, table, caption, tbody, tfoot, thead, tr, th, td, article, aside, canvas, details, embed, figure, figcaption, footer, header, hgroup, menu, nav, output, ruby, section, summary, time, mark, audio, video {
  margin: 0;
  padding: 0;
  border: 0;
}

/* BODY
=============================================================================*/

body {
  font-family: Helvetica, arial, freesans, clean, sans-serif;
  font-size: 16px;
  line-height: 1.6;
  color: #333;
  background-color: #fff;
  padding: 20px;
  max-width: 960px;
  margin: 0 auto;
}

@media screen and (max-width: 768px) {
  body {
  	font-size: 20px !important;
  }
}

body>*:first-child {
  margin-top: 0 !important;
}

body>*:last-child {
  margin-bottom: 0 !important;
}

/* BLOCKS
=============================================================================*/

p, blockquote, ul, ol, dl, table, pre {
  margin: 16px 0;
}

/* HEADERS
=============================================================================*/

h1, h2, h3, h4, h5, h6 {
  margin: 20px 0 10px;
  padding: 0;
  font-weight: bold;
  -webkit-font-smoothing: antialiased;
}

h1 tt, h1 code, h2 tt, h2 code, h3 tt, h3 code, h4 tt, h4 code, h5 tt, h5 code, h6 tt, h6 code {
  font-size: inherit;
}

h1 {
  font-size: 28px;
}

h1:first-of-type {
  font-size: 32px;
  text-align: center;
}

h2 {
  font-size: 26px;
}

h3 {
  font-size: 22px;
}

h4 {
  font-size: 18px;
}

h5 {
  font-size: 16px;
}

h6 {
  font-size: 14px;
}

body>h2:first-child, body>h1:first-child, body>h1:first-child+h2, body>h3:first-child, body>h4:first-child, body>h5:first-child, body>h6:first-child {
  margin-top: 0;
  padding-top: 0;
}

a:first-child h1, a:first-child h2, a:first-child h3, a:first-child h4, a:first-child h5, a:first-child h6 {
  margin-top: 0;
  padding-top: 0;
}

h1+p, h2+p, h3+p, h4+p, h5+p, h6+p {
  margin-top: 10px;
}

/* LINKS
=============================================================================*/

a {
  color: #4183C4;
  text-decoration: none;
}

a:hover {
  color: #d0782a;
  text-decoration: underline;
}

a:active {
  color: #bd4147;
}

/* LISTS
=============================================================================*/

ul, ol {
  padding-left: 30px;
}

ul li > :first-child, 
ol li > :first-child, 
ul li ul:first-of-type, 
ol li ol:first-of-type, 
ul li ol:first-of-type, 
ol li ul:first-of-type {
  margin-top: 0px;
  margin-bottom: 0px;
}

ul ul, ul ol, ol ol, ol ul {
  margin-bottom: 0;
}

dl {
  padding: 0;
}

dl dt {
  font-size: 14px;
  font-weight: bold;
  font-style: italic;
  padding: 0;
  margin: 15px 0 5px;
}

dl dt:first-child {
  padding: 0;
}

dl dt>:first-child {
  margin-top: 0px;
}

dl dt>:last-child {
  margin-bottom: 0px;
}

dl dd {
  margin: 0 0 15px;
  padding: 0 15px;
}

dl dd>:first-child {
  margin-top: 0px;
}

dl dd>:last-child {
  margin-bottom: 0px;
}

/* CODE
=============================================================================*/

pre, code, tt {
  font-family: Consolas, "Liberation Mono", Courier, monospace;
}

code, tt {
  margin: 0 0px;
  padding: 0px 3px;
  white-space: nowrap;
  border: 1px solid #eaeaea;
  background-color: #eee;
  color: #00d;
  border-radius: 3px;
}

pre>code {
  margin: 0;
  padding: 0;
  white-space: pre;
  border: none;
  background: transparent;
}

pre {
  background-color: #f8f8f8;
  border: 1px solid #ccc;
  line-height: 19px;
  overflow: auto;
  padding: 6px 10px;
  border-radius: 3px;
}

pre code, pre tt {
  background-color: transparent;
  border: none;
}

kbd {
    -moz-border-bottom-colors: none;
    -moz-border-left-colors: none;
    -moz-border-right-colors: none;
    -moz-border-top-colors: none;
    background-color: #DDDDDD;
    background-image: linear-gradient(#F1F1F1, #DDDDDD);
    background-repeat: repeat-x;
    border-color: #DDDDDD #CCCCCC #CCCCCC #DDDDDD;
    border-image: none;
    border-radius: 2px 2px 2px 2px;
    border-style: solid;
    border-width: 1px;
    font-family: "Helvetica Neue",Helvetica,Arial,sans-serif;
    line-height: 16px;
    padding: 1px 4px;
}

/* QUOTES
=============================================================================*/
blockquote {
  border-left: 4px solid #42b983;
  padding: 0 15px;
  color: #d1d9e1;
  background: #474949;
}

blockquote>:first-child {
  margin-top: 0px;
}

blockquote>:last-child {
  margin-bottom: 0px;
}

/* HORIZONTAL RULES
=============================================================================*/

hr {
  clear: both;
  margin: 15px 0;
  height: 0px;
  overflow: hidden;
  border: none;
  background: transparent;
  border-bottom: 4px solid #ddd;
  padding: 0;
}

/* TABLES
=============================================================================*/
table {
	margin: 0 auto;
	border-collapse: collapse;
	width: 100%;
	box-sizing: border-box;
	margin-bottom: 30px;
	
}

table th, table td {
  border: 1px solid #ccc;
  padding: 6px 13px; 
}

table th {
  font-weight: bold;
  text-align: center !important;
  background-color: #9ec68e;
}

table tr {
  border-top: 1px solid #ccc;
  background-color: #fff;
}

table tr:nth-child(2n) {
  background-color: #dff0d8;
}

/* IMAGES
=============================================================================*/
img {
  max-width: 100%;
}

p > img {
  display: table;
  margin: 0 auto;
}
</style>
<style type="text/css">
.highlight  { background: #ffffff; }
.highlight .c { color: #999988; font-style: italic } /* Comment */
.highlight .err { color: #a61717; background-color: #e3d2d2 } /* Error */
.highlight .k { font-weight: bold } /* Keyword */
.highlight .o { font-weight: bold } /* Operator */
.highlight .cm { color: #999988; font-style: italic } /* Comment.Multiline */
.highlight .cp { color: #999999; font-weight: bold } /* Comment.Preproc */
.highlight .c1 { color: #999988; font-style: italic } /* Comment.Single */
.highlight .cs { color: #999999; font-weight: bold; font-style: italic } /* Comment.Special */
.highlight .gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */
.highlight .gd .x { color: #000000; background-color: #ffaaaa } /* Generic.Deleted.Specific */
.highlight .ge { font-style: italic } /* Generic.Emph */
.highlight .gr { color: #aa0000 } /* Generic.Error */
.highlight .gh { color: #999999 } /* Generic.Heading */
.highlight .gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */
.highlight .gi .x { color: #000000; background-color: #aaffaa } /* Generic.Inserted.Specific */
.highlight .go { color: #888888 } /* Generic.Output */
.highlight .gp { color: #555555 } /* Generic.Prompt */
.highlight .gs { font-weight: bold } /* Generic.Strong */
.highlight .gu { color: #aaaaaa } /* Generic.Subheading */
.highlight .gt { color: #aa0000 } /* Generic.Traceback */
.highlight .kc { font-weight: bold } /* Keyword.Constant */
.highlight .kd { font-weight: bold } /* Keyword.Declaration */
.highlight .kp { font-weight: bold } /* Keyword.Pseudo */
.highlight .kr { font-weight: bold } /* Keyword.Reserved */
.highlight .kt { color: #445588; font-weight: bold } /* Keyword.Type */
.highlight .m { color: #009999 } /* Literal.Number */
.highlight .s { color: #d14 } /* Literal.String */
.highlight .na { color: #008080 } /* Name.Attribute */
.highlight .nb { color: #0086B3 } /* Name.Builtin */
.highlight .nc { color: #445588; font-weight: bold } /* Name.Class */
.highlight .no { color: #008080 } /* Name.Constant */
.highlight .ni { color: #800080 } /* Name.Entity */
.highlight .ne { color: #990000; font-weight: bold } /* Name.Exception */
.highlight .nf { color: #990000; font-weight: bold } /* Name.Function */
.highlight .nn { color: #555555 } /* Name.Namespace */
.highlight .nt { color: #000080 } /* Name.Tag */
.highlight .nv { color: #008080 } /* Name.Variable */
.highlight .ow { font-weight: bold } /* Operator.Word */
.highlight .w { color: #bbbbbb } /* Text.Whitespace */
.highlight .mf { color: #009999 } /* Literal.Number.Float */
.highlight .mh { color: #009999 } /* Literal.Number.Hex */
.highlight .mi { color: #009999 } /* Literal.Number.Integer */
.highlight .mo { color: #009999 } /* Literal.Number.Oct */
.highlight .sb { color: #d14 } /* Literal.String.Backtick */
.highlight .sc { color: #d14 } /* Literal.String.Char */
.highlight .sd { color: #d14 } /* Literal.String.Doc */
.highlight .s2 { color: #d14 } /* Literal.String.Double */
.highlight .se { color: #d14 } /* Literal.String.Escape */
.highlight .sh { color: #d14 } /* Literal.String.Heredoc */
.highlight .si { color: #d14 } /* Literal.String.Interpol */
.highlight .sx { color: #d14 } /* Literal.String.Other */
.highlight .sr { color: #009926 } /* Literal.String.Regex */
.highlight .s1 { color: #d14 } /* Literal.String.Single */
.highlight .ss { color: #990073 } /* Literal.String.Symbol */
.highlight .bp { color: #999999 } /* Name.Builtin.Pseudo */
.highlight .vc { color: #008080 } /* Name.Variable.Class */
.highlight .vg { color: #008080 } /* Name.Variable.Global */
.highlight .vi { color: #008080 } /* Name.Variable.Instance */
.highlight .il { color: #009999 } /* Literal.Number.Integer.Long */
.pl-c {
    color: #969896;
}

.pl-c1,.pl-mdh,.pl-mm,.pl-mp,.pl-mr,.pl-s1 .pl-v,.pl-s3,.pl-sc,.pl-sv {
    color: #0086b3;
}

.pl-e,.pl-en {
    color: #795da3;
}

.pl-s1 .pl-s2,.pl-smi,.pl-smp,.pl-stj,.pl-vo,.pl-vpf {
    color: #333;
}

.pl-ent {
    color: #63a35c;
}

.pl-k,.pl-s,.pl-st {
    color: #a71d5d;
}

.pl-pds,.pl-s1,.pl-s1 .pl-pse .pl-s2,.pl-sr,.pl-sr .pl-cce,.pl-sr .pl-sra,.pl-sr .pl-sre,.pl-src,.pl-v {
    color: #df5000;
}

.pl-id {
    color: #b52a1d;
}

.pl-ii {
    background-color: #b52a1d;
    color: #f8f8f8;
}

.pl-sr .pl-cce {
    color: #63a35c;
    font-weight: bold;
}

.pl-ml {
    color: #693a17;
}

.pl-mh,.pl-mh .pl-en,.pl-ms {
    color: #1d3e81;
    font-weight: bold;
}

.pl-mq {
    color: #008080;
}

.pl-mi {
    color: #333;
    font-style: italic;
}

.pl-mb {
    color: #333;
    font-weight: bold;
}

.pl-md,.pl-mdhf {
    background-color: #ffecec;
    color: #bd2c00;
}

.pl-mdht,.pl-mi1 {
    background-color: #eaffea;
    color: #55a532;
}

.pl-mdr {
    color: #795da3;
    font-weight: bold;
}

.pl-mo {
    color: #1d3e81;
}
.task-list {
padding-left:10px;
margin-bottom:0;
}

.task-list li {
    margin-left: 20px;
}

.task-list-item {
list-style-type:none;
padding-left:10px;
}

.task-list-item label {
font-weight:400;
}

.task-list-item.enabled label {
cursor:pointer;
}

.task-list-item+.task-list-item {
margin-top:3px;
}

.task-list-item-checkbox {
display:inline-block;
margin-left:-20px;
margin-right:3px;
vertical-align:1px;
}
</style>
<base target=_blank>
</head>
<body>
<h1 id="-6-">第6章：快速响应的用户界面</h1>
<h2 id="-ui-p107">浏览器UI线程P107</h2>
<h3 id="-">浏览器限制</h3>
<p>浏览器限制了JS任务的运行时间，此类限制分两种：调用栈大小限制和长时间运行脚本限制。长时间运行脚本被称为长时间运行脚本定时器或失控脚本定时器。第一种方法是记录自脚本开始以来执行的语句的数量；第二种是记录脚本执行的总时长。</p>
<blockquote>
<p>如果界面在100ms内响应用户输入，用户会认为自己在直接操作界面中的对象。超过100ms意味着用户会感觉到自己与界面失去联系。</p>
</blockquote>
<h2 id="-">使用定时器让出时间片段</h2>
<h3 id="-">定时器基础</h3>
<p>在JS中使用<code>setTimeout()</code>和<code>setInterval()</code>创建定时器，它们接收相同的参数：要执行的函数和执行前的等待时间（<code>ms</code>）。<code>setTimeout()</code>函数创建了一个只执行一次的定时器，而<code>setInterval()</code>创建了一个周期性重复运行的定时器。</p>
<blockquote>
<p>定时器代码只有在创建它的函数执行完成之后，才有可能被执行。定时器代码会重置所有相关的浏览器限制，包括长时间运行脚本定时器。<strong>此外，调用栈也在定时器的代码中重置为<code>0</code>。</strong>这一特性使得定时器成为长时间运行JS代码理想的跨浏览器解决方案。</p>
</blockquote>
<h3 id="-">定时器精度</h3>
<p>JS定时器延迟通常不太精准，相差大约几毫秒。定时器延时的最小值有助于避免在其他浏览器和其他操作系统中的定时器出现分辨率问题。</p>
<h3 id="-">使用定时器处理数组</h3>
<p>使用定时器取代循环的两个决定性因素：</p>
<ul>
<li>处理过程不需要同步？</li><li>数据不需要按顺序处理？</li></ul>
<p>基本的异步代码模式：</p>
<pre><code>items = [&quot;a&quot;, &quot;b&quot;];
var todo = items.concat();//克隆原数组
setTimeout(function () {
    //取得数组的下个元素并进行处理
    process(todo.shift());

    //如果还有需要处理的元素，创建另一个定时器
    if (todo.length &gt; 0) {
        setTimeout(arguments.callee, 25);
    } else {
        callback(items)
    }
});
</code></pre><h3 id="-p116">分割任务P116</h3>
<h3 id="-">记录代码运行时间</h3>
<p>通过原生的<code>Date</code>对象来跟踪代码的运行时间。每个新创建的<code>Date</code>对象以当前系统时间初始化，可以定时创建<code>Date</code>对象并比较它们的值来记录代码运行时间。</p>
<h3 id="-">定时器与性能</h3>
<p>在Web应用中限制高频率重复定时器的数量，创建一个独立的重复定时器，每次执行多个操作。</p>
<h2 id="web-workers">Web Workers</h2>
<p>Web Workers API引入了一个接口，能使代码运行且不占用浏览器UI线程的时间。</p>
<h3 id="worder-">Worder运行环境</h3>
<p>Web Workers没有绑定UI线程，它们不能访问浏览器的许多资源。Web Workers从外部线程修改DOM会导致用户界面出现错误，每个Web Workers都有自己的全局运行环境，其功能只是JS特性的一个子集。</p>
<p>Web Workers运行环境：</p>
<ul>
<li>一个<code>navigator</code>对象，只包含四个属性：<code>appName</code>、<code>appVersion</code>、<code>user Agent</code>和<code>platform</code>。</li><li>一个<code>location</code>对象</li><li>一个<code>self</code>对象，指向全局worker对象</li><li>一个<code>importScripts()</code>方法，用来加载Worker所用到的外部JS文件</li><li>所有的ES6对象，比如：<code>Object</code>、<code>Array</code>、<code>Date</code></li><li><code>XMLHTTPRequest</code>构造器</li><li><code>setTimeout()</code>和<code>setInterval()</code>方法</li><li>一个<code>close()</code>方法，它能立刻停止Worker运行</li></ul>
<p>Web Workers有着不同的全局运行环境，因此无法从JS代码中创建，需要创建一个完全独立的JS文件，其中包含了需要在Worker中运行的代码。</p>
<p>要创建人工线程，必须传入JS文件的URL：</p>
<pre><code>var worker = new Worker(&#39;code.js&#39;);
</code></pre><p>一旦代码执行，将为这个文件创建一个新的线程和一个新的运行环境。该文件会被异步下载，直到文件下载并执行完成后才会启动此Worker。</p>
<h3 id="-worker-">与Worker通信</h3>
<p>Worker与网页代码通过事件接口进行通信。网页代码可以通过<code>postMessage()</code>方法给Worker传递数据，它接收一个参数，即需要传递Worker的数据。此外，Worker还有一个用来接收信息的<code>onmessage</code>事件处理器。</p>
<pre><code>var worker = new Worker(&#39;code.js&#39;);
worker.onmessage = function (event) {
    console.log(event.data);
};
worker.postMessage(&quot;Andy&quot;);
</code></pre><h3 id="-">加载外部文件</h3>
<p>Worker通过<code>importScript()</code>方法加载外部JS文件，该方法接收一个或多个JS文件URL作为参数。<code>importScript()</code>的调用过程是阻塞式的，直到所有文件加载并执行完成后，脚本才会继续运行。由于Worker在UI线程之外运行，所以这种阻塞不会影响UI响应。</p>
<h3 id="-">实际应用</h3>
<p>Web Workers适用于处理纯数据，或者与浏览器UI无关的长时间运行脚本。</p>
<p>解析一个大字符串只是许多受益于Web Workers的任务之一。其他受益的任务：</p>
<ul>
<li>编码/解码大字符串</li><li>复杂数学运算（包括图像或视频处理）</li><li>大数组排序</li></ul>
<h2 id="-">小结</h2>
<p>高效管理UI线程就是要确保JS不能运行太长时间。</p>
<ul>
<li>任何JS任务都不应当执行超过100Ms。过长的运行时间会导致UI更新出现明显的延迟，从而对用户体验产生负面影响。</li><li>JS运行期间，浏览器响应用户交互的行为存在差异</li><li>定时器可用来安排代码延迟执行，可以把长时间运行的脚本分解成一系列的小任务</li><li>Web Workers允许在UI线程外部执行JS代码，从而避免锁定UI</li></ul>

</body>
</html>
<!-- This document was created with MarkdownPad, the Markdown editor for Windows (http://markdownpad.com) -->
